

# 生成图片验证码
# 请求路径:/passport/image_code
#请求方式：get
# 请求参数 随机字符串
# 返回值 图片验证码
import random
from datetime import datetime

from flask import current_app, jsonify
from flask import json
from flask import request
import re

from flask import session

from info import constants, db
from info import redis_store
from info.libs.yuntongxun.sms import CCP
from info.models import User

from info.utils.captcha.captcha import captcha
from info.utils.response_code import RET
from  . import passport_blue

@passport_blue.route('/image_code',)
def image_code():


    data = request.args
    cur_id = data.get('cur_id')
    pre_id = data.get('pre_id')
    # 生成图片验证码
    _,text,image_data = captcha.generate_captcha()
    try:
        # 保存图片验证码到redis_store
        # redis_store.set("image_code:%s"%cur_id,text,constants.IMAGE_CODE_REDIS_EXPIRES)
        redis_store.set("image_code:%s" % cur_id, text, constants.IMAGE_CODE_REDIS_EXPIRES)

        # 判断是否有上一个图片验证码编号
        if pre_id:
            redis_store.delete("image_code:%s" % pre_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="redis操作异常")


    # 返回
    return image_data


# 请求路径: /passport/sms_code
# 请求参数 手机号 图片验证码 验证码编号
# 请求方式POST
#返回值 返回相应状态

@passport_blue.route('/sms_code', methods=['POST'])
def sms_code():
    """
        思路分析:
        1.接收参数
        2.校验参数,为空校验
        3.校验手机号格式是否正确
        4.通过验证码编号,取出图片验证码A
        5.判断验证码A是否过期
        6.判断图片验证码是否正确
        7.调用CCP发送短信方法
        8.判断短信是否发送成功
        9.保存短信到redis中
        10.返回发送的状态
        :return:
        """

    # 接收参数
    json_data = request.data
    dict_data = json.loads(json_data)
    mobile = dict_data.get("mobile")
    image_code = dict_data.get("image_code")
    image_code_id = dict_data.get("image_code_id")

    #2.校验参数,为空校验
    if not all([mobile,image_code,image_code_id]):
        return jsonify(errno=RET.PARAMERR,errmsg="参数不完整")

    # 验证手机号是否正确

    if not re.match("1[345678]\\d{9}",mobile):
        return jsonify(errno=RET.DATAERR, errmsg="手机号格式有误")

    # 通过验证码编号 取出图片验证码是否正确

    try:
        redis_image_code = redis_store.get("image_code:%s"%image_code_id)

    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="redis数据取出错误")

    # 判断验证码是否过期
    if not redis_image_code:
        return jsonify(errno=RET.NODATA,errmsg="验证码过期")

    # 判断验证码是否正确
    if image_code.upper() != redis_image_code.upper():
        return jsonify(errno=RET.DATAERR,errmsg="验证码填写错误")

    #调用CCP发送短信方法、
    sms_code = "%06d"%random.randint(0,999999)
    current_app.logger.error("短信验证码是 = %s" % sms_code)
    # try:
    #     ccp =CCP()
    #     result=ccp.send_template_sms(mobile,[sms_code,constants.SMS_CODE_REDIS_EXPIRES/60],1)
    #
    #
    # except Exception as e:
    #     current_app.logger.error(e)
    #     return jsonify(errno=RET.THIRDERR,errmsg="云通讯发送异常")

    # # 8.判断短信是否发送成功
    # if result == -1:
    #     return jsonify(errno=RET.DATAERR,errmsg="短信发送失败")



    # 保存短信到redis
    try:
        redis_store.set("sms_code:%s"%mobile,sms_code,constants.SMS_CODE_REDIS_EXPIRES)

    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg="短信验证码保存异常")

    # 返发送的状态
    return jsonify(errno=RET.OK,errmsg="短信发送成功")

# 注册用户
# 请求路径 /passport/register
# 请求方式 POST
# 请求参数 手机号mobile 短信验证码sms_code 密码password
# 返回值 errno errmsg

@passport_blue.route('/register', methods=['POST'])
def register():
    """
       1.获取参数
       2.校验参数,为空校验
       3.手机号格式校验
       4.根据手机号取出redis中的短信验证码
       5.判断短信验证码是否过期
       6.校验短信验证码是否正确
       7.创建用户对象
       8.设置用户对象的属性信息
       9.保存用户到数据库
       10.返回响应信息
       :return:
       """
    #获取参数
    json_data = request.data
    dict_data = json.loads(json_data)
    mobile = dict_data.get("mobile")
    sms_code = dict_data.get("sms_code")
    password = dict_data.get("password")

    # 2.校验参数,为空校验
    if not all([mobile,sms_code,password]):
        return jsonify(errno=RET.PARAMERR,errmsg="参数不完整")

    # 手机号格式校验
    if not re.match('1[345678]\\d{9}', mobile):
        return jsonify(errno=RET.DATAERR, errmsg="手机号格式有误")

    # 根据手机号取出redis中的短信验证码
    try:
        redis_sms_code = redis_store.get("sms_code:%s"%mobile)

    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg="获取短信验证码失败")


    # 判断短信验证码是否过期

    if not redis_sms_code:
        return jsonify(errno=RET.NODATA,errmsg="短信验证码已过期")


    # 校验短信验证码是否正确

    if redis_sms_code != sms_code:
        return jsonify(errno=RET.DATAERR,errmsg="短信验证码填写错误e")


    # 创建用户对象
    user = User()


    # 设置用户对象的属性信息
    user.nick_name = mobile
    user.password = password
    user.mobile = mobile


    # 保存用户到数据库
    try:
        db.session.add(user)
        db.session.commit()

    except Exception as e:
        current_app.logger.error(e)
        db.session.rollback()
        return jsonify(errno=RET.DBERR, errmsg="用户创建失败")

        # 10.返回响应信息
    return jsonify(errno=RET.OK, errmsg="注册成功")


#登陆用户
# 请求路径: /passport/login
# 请求方式: POST
# 请求参数: mobile,password
# 返回值: errno, errmsg

@passport_blue.route('/login', methods=['POST'])
def login():
    """
       思路分析:
       1.获取参数
       2.校验参数
       3.根据手机号取出用户对象
       4.判断密码正确性
       5.将用户的登陆信息存到session
       6.返回响应
       :return:
       """
    #获取参数
    json_data = request.data
    dict_data = json.loads(json_data)

    mobile = dict_data.get('mobile')
    password = dict_data.get("password")

    # 校验参数
    if not all([mobile,password]):
        return jsonify(errno=RET.PARAMERR,errmsg="参数不完整")


    # 根据手机号取出用户对象

    try:
        user = User.query.filter(User.mobile == mobile).first()
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg="查询用户失败")


    # 判断密码正确性
    if not user.check_passowrd(password):
        return jsonify(errno=RET.DATAERR,errmsg="密码输入错误")


    # 将用户的登陆信息存到session

    session["user_id"]=user.id
    session["nick_name"] = user.nick_name
    session["mobile"] = user.mobile

    user.last_login = datetime.now()

    # db.session.commit()操作可以省略，因为在配置文件中设置了自动提交

    # 6.返回响应
    return jsonify(errno=RET.OK, errmsg="登陆成功")

# 退出
#请求路径：/passport/logout
#请求方式 DELETE
# 返回值errno errmsg

@passport_blue.route('/logout', methods=['DELETE'])
def logout():
    """
    清除session

    :return:
    """
    session.pop("user_id",None)
    session.pop("nick_name",None)
    session.pop("mobile",None)

    return jsonify(errno=RET.OK,errmsg="退出成功")

